package com.skh.string;

import java.util.ArrayList;
import java.util.List;

/**
 * @Author: skh
 * @Date: 2020/3/20 10:32
 * @Description: 括号生成
 */
public class GenerateParenthesis {

    /**
     * 给出 n 代表生成括号的对数，请你写出一个函数，使其能够生成所有可能的并且有效的括号组合。
     *
     * 例如，给出 n = 3，生成结果为：
     *
     * [
     *   "((()))",
     *   "(()())",
     *   "(())()",
     *   "()(())",
     *   "()()()"
     * ]
     */


    /**
     * 当前左右括号都有大于 0个可以使用的时候，才产生分支；
     * <p>
     * 产生左分支的时候，只看当前是否还有左括号可以使用；
     * <p>
     * 产生右分支的时候，还受到左分支的限制，右边剩余可以使用的括号数量一定得在严格大于左边剩余的数量的时候，才可以产生分支；
     * <p>
     * 在左边和右边剩余的括号数都等于 00 的时候结算。
     *
     * @param n
     * @return
     */
    public List<String> generateParenthesis(int n) {
        List<String> result = new ArrayList<>();
        dfs("", n, n, result);

        return result;
    }

    public void dfs(String cur, int left, int right, List<String> result) {
        if (left == 0 && right == 0) {
            result.add(cur);
            return;
        }

        //剪枝（左括号可以使用的个数严格大于右括号可以使用的个数，才剪枝，注意这个细节）
        if (left > right) {
            return;
        }

        if (left > 0) {
            dfs(cur + "(", left - 1, right, result);
        }

        if (right > 0) {
            dfs(cur + ")", left, right - 1, result);
        }
    }
}
